home *** CD-ROM | disk | FTP | other *** search
/ Internet Publisher's Toolbox 2.0 / Internet Publisher's Toolbox.iso / html / rtf2html / wrtf2htm / rtftohtm.c < prev    next >
Encoding:
C/C++ Source or Header  |  1994-11-05  |  22.3 KB  |  1,271 lines

  1. /*
  2.     rtftohtml -    RTF to HTML translator
  3.  
  4.             Initial Implementation by Chris Hector (cjh@cray.com)
  5.             
  6.             (c) 1994 by Chris Hector
  7.             May be freely distributed.
  8.  
  9.             This translator was based on rtfskel. 
  10.             Kudos to Paul DuBois for his work in developing the
  11.             rtf reader and skeleton code.
  12.  
  13. */
  14. # include    <stdio.h>
  15. # include    <stdlib.h>
  16. # define Edef 
  17. # include    "rtf.h"
  18. # include    "rtftohtml.h"
  19. static void    UnknownClass ();
  20. static void    GroupClass ();
  21. static void    ControlClass ();
  22.  
  23. static void    Destination ();
  24. static void    SpecialChar ();
  25. static void    DocAttr ();
  26. static void    SectAttr ();
  27. static void    TblAttr ();
  28. static void    ParAttr ();
  29. static void    CharAttr ();
  30. static void    PictAttr ();
  31. static void    NeXTGrAttr ();
  32. static void    FieldAttr ();
  33. static void    TOCAttr ();
  34. static void    PosAttr ();
  35. static void     PictAttr();
  36.  
  37. char *outputMapName="html-map";
  38. char *TransFile=    "html-trans";
  39.  
  40. /* Translator Options */
  41. int debug=0;
  42. int IMG=0;
  43. int ToC=1;
  44. int WriteGraf=1;
  45.  
  46. /* Input State Variables */
  47. # define    PSAtCol1    0
  48. # define    PSInCol1    1
  49. # define    PSAtCol2    2
  50. # define    PSInCol2    3
  51.  
  52. int IStyle_Chg=1;                /* Has input style changed? */
  53. int lastleft=0;
  54.  
  55. char *ParStyle="";                /* Paragraph Style Name */
  56. TStyle_typ TStyle;                /* Bitmask of Text Styles */
  57. char *TFont="";                /* Points to Font String */
  58. int TSize=0;                    /* Pointsize of Font */
  59. int inTable=0;            /* In a table? */
  60. int cellno=0;
  61. int firstcell=0;
  62. int lastcell=0;
  63. int celldidx;        /* index of the cell definition */
  64. int ToCLev;                    /* Table of Contents Level */
  65. int CurFNote;
  66.  
  67.  
  68. int destination=rtfMaxDestination;                /* The current input destination */
  69. RTFFont        *fp;                /* Actual Font Pointer */
  70. int    OptDestFound;            /* Set if an optional destination was encountered */
  71.  
  72. InStateStack  *ISS=NULL;
  73.  
  74.  
  75. char PFileExt[16]="gif";    /* extension to be used for links 
  76.                             to pictures */
  77. char *FPrefix;        /* Base name of output files */
  78. char *FPrefixR;        /* Relative name of output files */
  79.  
  80. char *OutfileName;            /* output file name */
  81. int linkself;            /* set for internal links */
  82.  
  83. char    *outMap[rtfSC_MaxChar];
  84.  
  85. char *filename;
  86. static void ObjAttr ();
  87.  
  88.  
  89. /*
  90.  * Once-only writer initialization
  91.  */
  92.  
  93. void
  94. WriterInit ()
  95. {
  96.     FPrefix = RTFAlloc (256);
  97.     FPrefixR = RTFAlloc (256);
  98.     OutfileName = RTFAlloc (256);
  99.     filename = RTFAlloc (256);
  100.     if (FPrefix == (char *) NULL
  101.         || FPrefixR == (char *) NULL
  102.         || OutfileName == (char *) NULL
  103.         || filename == (char *) NULL)
  104.         RTFPanic ("Cannot allocate filename buffers.");
  105.     FPrefix[0] = '\0';
  106.     FPrefixR[0] = '\0';
  107.     OutfileName[0] = '\0';
  108.     filename[0] = '\0';
  109.  
  110.     if (RTFReadOutputMap (outputMapName, outMap, 1) == 0)
  111.         RTFPanic ("Cannot read output map %s", outputMapName);
  112.  
  113.     if (TransInit (TransFile) == 0)
  114.         RTFPanic ("Cannot read Translation File %s", TransFile);
  115. }
  116.  
  117.  
  118. /*
  119.  * Set up to process a single file
  120.  */
  121.  
  122. int
  123. do_main(argc, argv)
  124. int    argc;
  125. char    **argv;
  126. {
  127.     int i;
  128.     char versid[256];
  129.  
  130.  
  131.         debug=0;
  132.         IMG=0;
  133.         ToC=1;
  134.         ToCLev=0;
  135.         IStyle_Chg=1;
  136.         ParStyle="";
  137.         TFont="";
  138.         WriteGraf=1;
  139.         TSize=0;
  140.         cellno=0;
  141.         inTable=0;
  142.         firstcell=0;
  143.         lastcell=0;
  144.         CurFNote=0;
  145.         destination=rtfMaxDestination;
  146.         ISS=NULL;
  147.         strcpy(PFileExt,"gif");
  148.         FPrefix[0]='\0';
  149.         OptDestFound=0;
  150.  
  151.         --argc;
  152.         ++argv;
  153.        while (argc > 0 && **argv == '-') {
  154.         if (strcmp ("-V", *argv) == 0){
  155.             sprintf(versid,"rtftohtml Version %s\n",PVers);
  156.             RTFMsg(versid);
  157.             return (0);
  158.         } else if (strcmp ("-i", *argv) == 0){
  159.             IMG=1;
  160.         } else if (strcmp ("-G", *argv) == 0){
  161.             WriteGraf=0;
  162.         } else if (strcmp ("-T", *argv) == 0){
  163.             ToC=0;
  164.         } else if (strcmp ("-o", *argv) == 0){
  165.                 if (argc < 2) {
  166.             return (-1);
  167.                 }
  168.                 --argc;
  169.                 ++argv;
  170.                 strcpy(OutfileName,*argv);
  171.                 /*
  172.                     if the -o option was specified, use the file name
  173.                     as a prefix (stripping off .html or .HTML if present.)
  174.                 */
  175.                 i=strlen(argv[0]);
  176.                 if(i>5&&(strcmp(&(argv[0][i-5]),".html")==0||
  177.                     strcmp(&(argv[0][i-5]),".HTML")==0)){
  178.                     i=i-5;
  179.                 }
  180.                 if(i>256)i=255;
  181.                 strncpy(FPrefix,argv[0],(size_t) i);
  182.                 FPrefix[i]=0;
  183.         } else if (strcmp ("-P", *argv) == 0){
  184.             if (argc < 2) {
  185.                 return (-1);
  186.             }
  187.             --argc;
  188.             ++argv;
  189.             strncpy(PFileExt,*argv,(size_t) 15);
  190.             PFileExt[15] = '\0';
  191.         } else {
  192.             RTFMsg ("Don't understand %s option.\n", *argv);
  193.             return (-1);
  194.         }
  195.         --argc;
  196.         ++argv;
  197.     }
  198.     if (argc > 0)
  199.     {
  200.         if (freopen (argv[0], "r", stdin) == NULL)
  201.         {
  202.             RTFMsg ("Can't open \"%s\"\n", argv[0]);
  203.             return (0);
  204.         }
  205.  
  206.                 /*
  207.                     if the -o option was not specified, use the file name
  208.                     as a prefix (stripping off .rtf or .RTF if present.)
  209.                 */  
  210.                 if(FPrefix[0]==0){
  211.                     i=strlen(argv[0]);
  212.                     if(i>4&&(strcmp(&(argv[0][i-4]),".rtf")==0||
  213.                         strcmp(&(argv[0][i-4]),".RTF")==0)){
  214.                         i=i-4;
  215.                     }   
  216.                     if(i>256)i=256;
  217.                     strncpy(FPrefix,argv[0],(size_t) i);
  218.                     FPrefix[i]=0;
  219.                     sprintf(OutfileName,"%s.html",FPrefix);
  220.                 }   
  221.     } else {
  222.         RTFMsg("No input file given\n");
  223.         return (0);
  224.     }
  225.  
  226.     strcpy(FPrefixR, Basename (FPrefix));
  227.     HTMLInit();
  228.     
  229.     /* install class callbacks */
  230.  
  231.     RTFSetClassCallback (rtfUnknown, UnknownClass);
  232.     RTFSetClassCallback (rtfGroup, GroupClass);
  233.     RTFSetClassCallback (rtfText, PutHTML);
  234.     RTFSetClassCallback (rtfControl, ControlClass);
  235.  
  236.     /* de-install default destination callbacks */
  237.     RTFSetDestinationCallback(rtfInfo, NULL);
  238.     RTFSetDestinationCallback(rtfPict, NULL);
  239.     RTFSetDestinationCallback(rtfObject, NULL);
  240.     RTFSetDestinationCallback(rtfColorTbl, NULL);
  241.  
  242.         /*               
  243.                 Process the input stream.  Make sure the first intoken is
  244.                 a "{" so a state push will occur before anything else
  245.                 (need to preserve state 0 intact for section, paragraph,
  246.                 character default restoration).
  247.         */               
  248.                          
  249.         (void) RTFGetToken ();
  250.         if (!RTFCheckCM (rtfGroup, rtfBeginGroup))
  251.         {                
  252.                 RTFMsg ("malformed rtf file - does not begin with \"{\"\n");
  253.                 return (0);
  254.         }
  255.         RTFRouteToken ();       /* send "{" through router */
  256.  
  257.     return (1);
  258. }
  259.  
  260.  
  261. /*
  262.     Token class callbacks
  263. */
  264.  
  265.  
  266. static void UnknownClass ()
  267. {
  268.     /* use this to print warnings about unknown tokens */
  269.     if(OptDestFound){
  270.         /* Weed out optional destinations. No Warning given */
  271.         RTFSkipGroup ();
  272.         PopIState();
  273.         return;
  274.     }
  275.     RTFMsg ("Unknown symbol \"%s\" near line %ld, position %d.\n",
  276.                 rtfTextBuf, rtfLineNum, rtfLinePos);
  277. }
  278.  
  279.  
  280. static void GroupClass ()
  281. {
  282.     IStyle_Chg=1;
  283.     switch (rtfMajor)
  284.     {
  285.     case rtfBeginGroup:
  286.         /* push state */
  287.         PushIState();
  288.         break;
  289.     case rtfEndGroup:
  290.         /* pop state */
  291.         PopIState();
  292.         break;
  293.     }
  294. }
  295.  
  296.  
  297. /*
  298.     Process control symbol.
  299. */
  300.  
  301. static void ControlClass ()
  302. {
  303.     switch (rtfMajor)
  304.     {
  305.     case rtfVersion:
  306.         break;
  307.     case rtfDefFont:
  308.         /*
  309.             rtfParam contains the default font number.  Beware
  310.             that there is no guarantee that this font will actually
  311.             be listed in the font table!
  312.         */
  313.         break;
  314.     case rtfCharSet:
  315.         /*CharSet ();*/
  316.         break;
  317.     case rtfDestination:
  318.         Destination ();
  319.         break;
  320.     case rtfFontFamily:
  321.         /* only occurs within font table - can ignore */
  322.         break;
  323.     case rtfColorName:
  324.         /* only occurs within color table - can ignore */
  325.         break;
  326.     case rtfSpecialChar:
  327.         SpecialChar ();
  328.         break;
  329.     case rtfStyleAttr:
  330.         /* only occurs within stylesheet - can ignore */
  331.         break;
  332.     case rtfDocAttr:
  333.         DocAttr ();
  334.         break;
  335.     case rtfSectAttr:
  336.         SectAttr ();
  337.         break;
  338.     case rtfTblAttr:
  339.         TblAttr ();
  340.         break;
  341.     case rtfObjAttr:
  342.         ObjAttr ();
  343.         break;
  344.     case rtfParAttr:
  345.         ParAttr ();
  346.         break;
  347.     case rtfCharAttr:
  348.         CharAttr ();
  349.         break;
  350.     case rtfPictAttr:
  351.         PictAttr ();
  352.         break;
  353.     case rtfNeXTGrAttr:
  354.         NeXTGrAttr ();
  355.         break;
  356.     case rtfFieldAttr:
  357.         FieldAttr ();
  358.         break;
  359.     case rtfTOCAttr:
  360.         TOCAttr ();
  361.         break;
  362.     case rtfPosAttr:
  363.         PosAttr ();
  364.         break;
  365.     }
  366. }
  367.  
  368.  
  369. /*
  370.     Control class major number handlers.  Each one switches on
  371.     the minor numbers that occur within the major number.
  372. */
  373.  
  374.  
  375. static void Destination ()
  376. {
  377.     destination=rtfMinor;
  378.     OptDestFound=0;
  379.     IStyle_Chg=1;
  380.     switch (rtfMinor)
  381.     {
  382.     case rtfPict:
  383.         break;
  384.     case rtfNeXTGraphic:
  385.         break;
  386.     case rtfFootnote:
  387.         firstcell=lastcell;
  388.         break;
  389.     case rtfHeader:
  390.     case rtfHeaderLeft:
  391.     case rtfHeaderRight:
  392.     case rtfHeaderFirst:
  393.     case rtfFooter:
  394.     case rtfFooterLeft:
  395.     case rtfFooterRight:
  396.     case rtfFooterFirst:
  397.         RTFSkipGroup ();
  398.         PopIState();
  399.         break;
  400.     case rtfFNSep:
  401.         RTFSkipGroup ();
  402.         PopIState();
  403.         break;
  404.     case rtfFNContSep:
  405.         RTFSkipGroup ();
  406.         PopIState();
  407.         break;
  408.     case rtfFNContNotice:
  409.         RTFSkipGroup ();
  410.         PopIState();
  411.         break;
  412.     case rtfInfo:
  413.         break;
  414.     case rtfStyleSheet:
  415.         /* will never occur because of default destination reader */
  416.         break;
  417.     case rtfFontTbl:
  418.         /* will never occur because of default destination reader */
  419.         break;
  420.     case rtfColorTbl:
  421.         RTFSkipGroup ();
  422.         PopIState();
  423.         break;
  424.     case rtfField:
  425.     case rtfFieldResult:
  426.         break;
  427.     case rtfFieldInst:
  428.         RTFSkipGroup ();
  429.         PopIState();
  430.         break;
  431.     case rtfIndex:
  432.     case rtfIndexBold:
  433.     case rtfIndexItalic:
  434.     case rtfIndexText:
  435.     case rtfIndexRange:
  436.         break;
  437.     case rtfTOC:
  438.         break;
  439.     case rtfBookmarkStart:
  440.         break;
  441.     case rtfBookmarkEnd:
  442.         break;
  443.     case rtfITitle:
  444.         break;
  445.     case rtfISubject:
  446.     case rtfIAuthor:
  447.     case rtfIOperator:
  448.     case rtfIKeywords:
  449.     case rtfIComment:
  450.     case rtfIVerscomm:
  451.     case rtfIDoccomm:
  452.         RTFSkipGroup ();
  453.         PopIState();
  454.         break;
  455.     case rtfObject:
  456.         linkself=0;
  457.         break;
  458.     case rtfObjItem:
  459.         break;
  460.     case rtfObjData:
  461.     case rtfObjClass:
  462.     case rtfObjName:
  463.     case rtfObjTopic:
  464.         RTFSkipGroup ();
  465.         PopIState();
  466.         break;
  467.     case rtfParNumText:
  468.         break;
  469.     case rtfObjResult:
  470.         break;
  471.     case rtfAnnotation:
  472.         break;
  473.     case rtfAnnotID:
  474.         break;
  475.     default:
  476.         RTFSkipGroup ();
  477.         PopIState();
  478.         break;        
  479.     }
  480.     HTMLStartDivert();
  481. }
  482.  
  483.  
  484. static void SpecialChar ()
  485. {
  486.     char fnote[8];
  487.     switch (rtfMinor)
  488.     {
  489.     case rtfCurHeadPage:
  490.         break;
  491.     case rtfCurFNote:
  492.         if(destination!=rtfFootnote)CurFNote++;
  493.         sprintf(fnote,"%d",CurFNote);
  494.         PutHTMLString(fnote);
  495.         break;
  496.     case rtfCurHeadPict:
  497.         break;
  498.     case rtfCurHeadDate:
  499.         break;
  500.     case rtfCurHeadTime:
  501.         break;
  502.     case rtfFormula:
  503.         break;
  504.     case rtfNoBrkSpace:
  505.         rtfMinor = rtfSC_nobrkspace;
  506.         PutHTML ();
  507.         break;
  508.     case rtfNoReqHyphen:
  509.         break;
  510.     case rtfNoBrkHyphen:
  511.         rtfMinor = rtfSC_nobrkhyphen;
  512.         PutHTML ();
  513.     case rtfRow:
  514.         if(lastcell==0){
  515.             lastcell=celldidx;
  516.             }
  517.         cellno=firstcell;
  518.         PutSpecial(rtfRow);
  519.         break;
  520.     case rtfSect:
  521.         break;
  522.     case rtfPage:
  523.         break;
  524.     case rtfLine:
  525.         rtfMinor = rtfSC_newline;
  526.         PutHTML ();
  527.         break;
  528.     case rtfPar:
  529.         PutSpecial(rtfPar);
  530.         break;
  531.     case rtfBullet:
  532.         rtfMinor = rtfSC_bullet;
  533.         PutHTML ();
  534.         break;
  535.     case rtfCell:
  536.         if(cellno++>=CELLMAX){
  537.             RTFMsg ("Overflowed the total number of cells %s\n",CELLMAX);
  538.             cellno--;
  539.         }
  540.         break;
  541.     case rtfTab:
  542.         PutSpecial(rtfTab);
  543.         break;
  544. /*
  545.     case rtfCurAnnot:
  546.         break;
  547. */
  548.     case rtfCurAnnotRef:
  549.         break;
  550.     case rtfFNoteSep:
  551.         break;
  552.     case rtfFNoteCont:
  553.         break;
  554.     case rtfColumn:
  555.         break;
  556.     case rtfOptDest:
  557.         OptDestFound=1;
  558.         break;
  559.     case rtfIIntVersion:
  560.         break;
  561.     case rtfICreateTime:
  562.         break;
  563.     case rtfIRevisionTime:
  564.         break;
  565.     case rtfIPrintTime:
  566.         break;
  567.     case rtfIBackupTime:
  568.         break;
  569.     case rtfIEditTime:
  570.         break;
  571.     case rtfIYear:
  572.         break;
  573.     case rtfIMonth:
  574.         break;
  575.     case rtfIDay:
  576.         break;
  577.     case rtfIHour:
  578.         break;
  579.     case rtfIMinute:
  580.         break;
  581.     case rtfINPages:
  582.         break;
  583.     case rtfINWords:
  584.         break;
  585.     case rtfINChars:
  586.         break;
  587.     case rtfIIntID:
  588.         break;
  589.     case rtfEmDash:
  590.         rtfMinor = rtfSC_emdash;
  591.         PutHTML ();
  592.         break;
  593.     case rtfEnDash:
  594.         rtfMinor = rtfSC_endash;
  595.         PutHTML ();
  596.         break;
  597.     case rtfLQuote:
  598.         rtfMinor = rtfSC_quoteleft;
  599.         PutHTML ();
  600.         break;
  601.     case rtfRQuote:
  602.         rtfMinor = rtfSC_quoteright;
  603.         PutHTML ();
  604.         break;
  605.     case rtfLDblQuote:
  606.         rtfMinor = rtfSC_quotedblleft;
  607.         PutHTML ();
  608.         break;
  609.     case rtfRDblQuote:
  610.         rtfMinor = rtfSC_quotedblright;
  611.         PutHTML ();
  612.         break;
  613.     }
  614. }
  615.  
  616.  
  617. static void DocAttr ()
  618. {
  619.     switch (rtfMinor)
  620.     {
  621.     case rtfPaperWidth:
  622.         break;
  623.     case rtfPaperHeight:
  624.         break;
  625.     case rtfLeftMargin:
  626.         break;
  627.     case rtfRightMargin:
  628.         break;
  629.     case rtfTopMargin:
  630.         break;
  631.     case rtfBottomMargin:
  632.         break;
  633.     case rtfFacingPage:
  634.         break;
  635.     case rtfGutterWid:
  636.         break;
  637.     case rtfDefTab:
  638.         break;
  639.     case rtfWidowCtrl:
  640.         break;
  641.     case rtfHyphHotZone:
  642.         break;
  643.     case rtfFNoteEndSect:
  644.         break;
  645.     case rtfFNoteEndDoc:
  646.         break;
  647.     case rtfFNoteText:
  648.         break;
  649.     case rtfFNoteBottom:
  650.         break;
  651.     case rtfFNoteStart:
  652.         break;
  653.     case rtfFNoteRestart:
  654.         break;
  655.     case rtfPageStart:
  656.         break;
  657.     case rtfLineStart:
  658.         break;
  659.     case rtfLandscape:
  660.         break;
  661.     case rtfFracWidth:
  662.         break;
  663.     case rtfNextFile:
  664.         break;
  665.     case rtfRTFDefault:
  666.         break;
  667.     case rtfRevisions:
  668.         break;
  669.     case rtfMirrorMargin:
  670.         break;
  671.     case rtfRevDisplay:
  672.         break;
  673.     case rtfRevBar:
  674.         break;
  675.     }
  676. }
  677.  
  678.  
  679. static void SectAttr ()
  680. {
  681.     switch (rtfMinor)
  682.     {
  683.     case rtfSectDef:
  684.         break;
  685.     case rtfNoBreak:
  686.         break;
  687.     case rtfColBreak:
  688.         break;
  689.     case rtfPageBreak:
  690.         break;
  691.     case rtfEvenBreak:
  692.         break;
  693.     case rtfOddBreak:
  694.         break;
  695.     case rtfPageStarts:
  696.         break;
  697.     case rtfPageCont:
  698.         break;
  699.     case rtfPageRestart:
  700.         break;
  701.     case rtfPageDecimal:
  702.         break;
  703.     case rtfPageURoman:
  704.         break;
  705.     case rtfPageLRoman:
  706.         break;
  707.     case rtfPageULetter:
  708.         break;
  709.     case rtfPageLLetter:
  710.         break;
  711.     case rtfPageNumRight:
  712.         break;
  713.     case rtfPageNumTop:
  714.         break;
  715.     case rtfHeaderY:
  716.         break;
  717.     case rtfFooterY:
  718.         break;
  719.     case rtfLineModulus:
  720.         break;
  721.     case rtfLineDist:
  722.         break;
  723.     case rtfLineStarts:
  724.         break;
  725.     case rtfLineRestart:
  726.         break;
  727.     case rtfLineRestartPg:
  728.         break;
  729.     case rtfLineCont:
  730.         break;
  731.     case rtfTopVAlign:
  732.         break;
  733.     case rtfBottomVAlign:
  734.         break;
  735.     case rtfCenterVAlign:
  736.         break;
  737.     case rtfJustVAlign:
  738.         break;
  739.     case rtfColumns:
  740.         break;
  741.     case rtfColumnSpace:
  742.         break;
  743.     case rtfColumnLine:
  744.         break;
  745.     case rtfENoteHere:
  746.         break;
  747.     case rtfTitleSpecial:
  748.         break;
  749.     }
  750. }
  751.  
  752. static void ObjAttr ()
  753. {
  754.     switch (rtfMinor)
  755.     {
  756.     case rtfObjLinkSelf:
  757.         linkself=1;
  758.         break;
  759.     }
  760. }
  761. static void TblAttr ()
  762. {
  763.     switch (rtfMinor)
  764.     {
  765.     case rtfCellBordBottom:
  766.         break;
  767.     case rtfCellBordTop:
  768.         break;
  769.     case rtfCellBordLeft:
  770.         break;
  771.     case rtfCellBordRight:
  772.         break;
  773.     case rtfRowDef:
  774.         lastcell=0;
  775.         cellno=firstcell;
  776.         celldidx=firstcell;
  777.         cell[celldidx].merge=0;
  778.         cell[celldidx].just=AdjLPad;
  779.         lastleft=0;
  780.         break;
  781.     case rtfRowLeft:
  782.         break;
  783.     case rtfRowRight:
  784.         break;
  785.     case rtfRowCenter:
  786.         break;
  787.     case rtfRowGapH:
  788.         break;
  789.     case rtfRowHt:
  790.         break;
  791.     case rtfRowLeftEdge:
  792.         lastleft=rtfParam;
  793.         break;
  794.     case rtfCellPos:
  795.         cell[celldidx].width=(rtfParam-lastleft)/TWIPSperCHAR;
  796.         if(cell[celldidx].width<=1)cell[celldidx].width=2;
  797.         lastleft=rtfParam;
  798.         if(celldidx++>=CELLMAX){
  799.             RTFMsg ("Overflowed the total number of cells %s\n",CELLMAX);
  800.             celldidx--;
  801.         }
  802.         cell[celldidx].just=AdjLPad;
  803.         cell[celldidx].merge=0;
  804.         break;
  805.     case rtfMergeRngFirst:
  806.         break;
  807.     case rtfMergePrevious:
  808.         cell[celldidx].merge=1;
  809.         break;
  810.     }
  811. }
  812. int parnest=0;
  813.  
  814. static void ParAttr ()
  815. {
  816.     RTFStyle *newstyle;
  817.     int i;
  818.     IStyle_Chg=1;
  819.     switch (rtfMinor)
  820.     {
  821.     case rtfParDef:    /* set the input paragraph stype to null */
  822.         ParStyle="";    
  823.         TStyle = stylePlain;
  824.         inTable=0;
  825.         break;
  826.     case rtfStyleNum:
  827.         /* When we call RTFExpandStyle, ParAttr will get recursively
  828.         called. We only perform processing on the outermost style. */
  829.         if(parnest==0){
  830.             newstyle=RTFGetStyle(rtfParam);
  831.             if (newstyle == (RTFStyle *) NULL)
  832.                 break;
  833.             ParStyle=newstyle->rtfSName;
  834.                 parnest++;
  835.                 RTFExpandStyle(rtfParam);
  836.                 parnest--;
  837.         }
  838.         break;
  839.     case rtfQuadLeft:
  840.         cell[cellno].just=AdjLPad;
  841.         break;
  842.     case rtfQuadRight:
  843.         cell[cellno].just=AdjRPad;
  844.         break;
  845.     case rtfQuadJust:
  846.         cell[cellno].just=AdjLPad;
  847.         break;
  848.     case rtfQuadCenter:
  849.         cell[cellno].just=AdjCpad;
  850.         break;
  851.     case rtfFirstIndent:
  852.         break;
  853.     case rtfLeftIndent:
  854.         break;
  855.     case rtfRightIndent:
  856.         break;
  857.     case rtfSpaceBefore:
  858.         break;
  859.     case rtfSpaceAfter:
  860.         break;
  861.     case rtfSpaceBetween:
  862.         break;
  863.     case rtfInTable:
  864.         inTable=1;
  865.         break;
  866.     case rtfKeep:
  867.         break;
  868.     case rtfKeepNext:
  869.         break;
  870.     case rtfSideBySide:
  871.         break;
  872.     case rtfPBBefore:
  873.         break;
  874.     case rtfNoLineNum:
  875.         break;
  876.     case rtfTabPos:
  877.         break;
  878.     case rtfTabRight:
  879.         break;
  880.     case rtfTabCenter:
  881.         break;
  882.     case rtfTabDecimal:
  883.         break;
  884.     case rtfTabBar:
  885.         break;
  886.     case rtfBorderTop:
  887.         break;
  888.     case rtfBorderBottom:
  889.         break;
  890.     case rtfBorderLeft:
  891.         break;
  892.     case rtfBorderRight:
  893.         break;
  894.     case rtfBorderBar:
  895.         break;
  896.     case rtfBorderBox:
  897.         break;
  898.     case rtfBorderBetween:
  899.         break;
  900.     case rtfBorderSingle:
  901.         break;
  902.     case rtfBorderThick:
  903.         break;
  904.     case rtfBorderShadow:
  905.         break;
  906.     case rtfBorderDouble:
  907.         break;
  908.     case rtfBorderDot:
  909.         break;
  910.     case rtfBorderHair:
  911.         break;
  912.     case rtfBorderSpace:
  913.         break;
  914.     case rtfLeaderDot:
  915.         break;
  916.     case rtfLeaderHyphen:
  917.         break;
  918.     case rtfLeaderUnder:
  919.         break;
  920.     case rtfLeaderThick:
  921.         break;
  922.     }
  923. }
  924.  
  925.  
  926. /*
  927.     Several of the attributes can be turned off with param value
  928.     of zero (e.g., \b vs. \b0).
  929. */
  930.  
  931. static void CharAttr ()
  932. {
  933. RTFFont        *fp;
  934. int    turnOn = (rtfParam != 0);
  935.     IStyle_Chg=1;
  936.     switch (rtfMinor)
  937.     {
  938.     case rtfPlain:
  939.         TStyle = stylePlain;
  940.         break;
  941.     case rtfBold:
  942.         if(turnOn){
  943.             TStyle |= styleBold;
  944.         } else {
  945.             TStyle &= ~styleBold;
  946.         }
  947.         break;
  948.     case rtfItalic:
  949.         if(turnOn){
  950.             TStyle |= styleItalic;
  951.         } else {
  952.             TStyle &= ~styleItalic;
  953.         }
  954.         break;
  955.     case rtfStrikeThru:
  956.         if(turnOn){
  957.             TStyle |= styleStrikeThru;
  958.         } else {
  959.             TStyle &= ~styleStrikeThru;
  960.         }
  961.         break;
  962.     case rtfOutline:
  963.         if(turnOn){
  964.             TStyle |= styleOutline;
  965.         } else {
  966.             TStyle &= ~styleOutline;
  967.         }
  968.         break;
  969.     case rtfShadow:
  970.         if(turnOn){
  971.             TStyle |= styleShadow;
  972.         } else {
  973.             TStyle &= ~styleShadow;
  974.         }
  975.         break;
  976.     case rtfSmallCaps:
  977.         if(turnOn){
  978.             TStyle |= styleSmallCaps;
  979.         } else {
  980.             TStyle &= ~styleSmallCaps;
  981.         }
  982.         break;
  983.     case rtfAllCaps:
  984.         if(turnOn){
  985.             TStyle |= styleAllCaps;
  986.         } else {
  987.             TStyle &= ~styleAllCaps;
  988.         }
  989.         break;
  990.     case rtfInvisible:
  991.         if(turnOn){
  992.             TStyle |= styleInvisible;
  993.         } else {
  994.             TStyle &= ~styleInvisible;
  995.         }
  996.         break;
  997.     case rtfFontNum:
  998.         if((fp = RTFGetFont(rtfParam)) == (RTFFont *) NULL){
  999.             TFont="";
  1000.         } else {
  1001.             TFont=fp->rtfFName;
  1002.         }
  1003.         break;
  1004.     case rtfFontSize:
  1005.         TSize=rtfParam;
  1006.         break;
  1007.     case rtfExpand:
  1008.         break;
  1009.     case rtfUnderline:
  1010.         if(turnOn){
  1011.             TStyle |= styleUnderline;
  1012.         } else {
  1013.             TStyle &= ~styleUnderline;
  1014.         }
  1015.         break;
  1016.     case rtfWordUnderline:
  1017.         if(turnOn){
  1018.             TStyle |= styleWUnderline;
  1019.         } else {
  1020.             TStyle &= ~styleWUnderline;
  1021.         }
  1022.         break;
  1023.     case rtfDotUnderline:
  1024.         if(turnOn){
  1025.             TStyle |= styleDUnderline;
  1026.         } else {
  1027.             TStyle &= ~styleDUnderline;
  1028.         }
  1029.         break;
  1030.     case rtfDbUnderline:
  1031.         if(turnOn){
  1032.             TStyle |= styleDbUnderline;
  1033.         } else {
  1034.             TStyle &= ~styleDbUnderline;
  1035.         }
  1036.         break;
  1037.     case rtfNoUnderline:
  1038.         TStyle&=~(styleUnderline|styleWUnderline|
  1039.                 styleDUnderline|styleDbUnderline);
  1040.         break;
  1041.     case rtfSuperScript:
  1042.         if(turnOn){
  1043.             TStyle |= styleSuperScript;
  1044.         } else {
  1045.             TStyle &= ~styleSuperScript;
  1046.         }
  1047.         break;
  1048.     case rtfSubScript:
  1049.         if(turnOn){
  1050.             TStyle |= styleSubScript;
  1051.         } else {
  1052.             TStyle &= ~styleSubScript;
  1053.         }
  1054.         break;
  1055.     case rtfRevised:
  1056.         break;
  1057.     case rtfForeColor:
  1058.         break;
  1059.     case rtfBackColor:
  1060.         break;
  1061.     }
  1062. }
  1063.  
  1064. static void PictAttr ()
  1065. {
  1066.     char *ptr,LinkBuf[512];
  1067.     switch (rtfMinor)
  1068.     {
  1069.     case rtfMacQD:
  1070.         PictExt="pict";
  1071.         PictType=FTPICT;
  1072.         RTFSetClassCallback (rtfText, NabPicture);
  1073.         break;
  1074.     case rtfWinMetafile:
  1075.         PictExt="wmf";
  1076.         PictType=FTWMF;
  1077.         RTFSetClassCallback (rtfText, NabPicture);
  1078.         break;
  1079.     case rtfWinBitmap:
  1080.         PictExt="bmp";
  1081.         PictType=FTBMF;
  1082.         RTFSetClassCallback (rtfText, NabPicture);
  1083.         break;
  1084.     case rtfPicWid:
  1085.         break;
  1086.     case rtfPicHt:
  1087.         break;
  1088.     case rtfPicGoalWid:
  1089.         PicGoalWid=rtfParam;
  1090.         break;
  1091.     case rtfPicGoalHt:
  1092.         PicGoalHt=rtfParam;
  1093.         break;
  1094.     case rtfPicScaleX:
  1095.         break;
  1096.     case rtfPicScaleY:
  1097.         break;
  1098.     case rtfPicScaled:
  1099.         break;
  1100.     case rtfPicCropTop:
  1101.         break;
  1102.     case rtfPicCropBottom:
  1103.         break;
  1104.     case rtfPicCropLeft:
  1105.         break;
  1106.     case rtfPicCropRight:
  1107.         break;
  1108.     case rtfPixelBits:
  1109.         break;
  1110.     case rtfBitmapPlanes:
  1111.         break;
  1112.     case rtfBitmapWid:
  1113.         break;
  1114.     case rtfPicBinary:
  1115.         RTFSkipGroup ();        /* Not sure this will work on binary input data */
  1116.         PopIState();
  1117.         break;
  1118.     }
  1119. }
  1120.  
  1121.  
  1122. static void NeXTGrAttr ()
  1123. {
  1124.     switch (rtfMinor)
  1125.     {
  1126.     case rtfNeXTGWidth:
  1127.         break;
  1128.     case rtfNeXTGHeight:
  1129.         break;
  1130.     }
  1131. }
  1132.  
  1133.  
  1134. static void FieldAttr ()
  1135. {
  1136.     switch (rtfMinor)
  1137.     {
  1138.     case rtfFieldDirty:
  1139.         break;
  1140.     case rtfFieldEdited:
  1141.         break;
  1142.     case rtfFieldLocked:
  1143.         break;
  1144.     case rtfFieldPrivate:
  1145.         break;
  1146.     }
  1147. }
  1148.  
  1149.  
  1150. static void TOCAttr ()
  1151. {
  1152.     switch (rtfMinor)
  1153.     {
  1154.     case rtfTOCType:
  1155.         break;
  1156.     case rtfTOCLevel:
  1157.         ToCLev=rtfParam;
  1158.         break;
  1159.     }
  1160. }
  1161.  
  1162.  
  1163. static void PosAttr ()
  1164. {
  1165.     switch (rtfMinor)
  1166.     {
  1167.     case rtfPosX:
  1168.         break;
  1169.     case rtfPosXCenter:
  1170.         break;
  1171.     case rtfPosXInside:
  1172.         break;
  1173.     case rtfPosXLeft:
  1174.         break;
  1175.     case rtfPosXOutSide:
  1176.         break;
  1177.     case rtfPosXRight:
  1178.         break;
  1179.     case rtfPosY:
  1180.         break;
  1181.     case rtfPosYInline:
  1182.         break;
  1183.     case rtfPosYTop:
  1184.         break;
  1185.     case rtfPosYCenter:
  1186.         break;
  1187.     case rtfPosYBottom:
  1188.         break;
  1189.     case rtfAbsWid:
  1190.         break;
  1191. /*
  1192.     case rtfTextDistX:
  1193.         break;
  1194. */
  1195.     case rtfTextDistY:
  1196.         break;
  1197.     case rtfRPosMargV:
  1198.         break;
  1199.     case rtfRPosPageV:
  1200.         break;
  1201.     case rtfRPosMargH:
  1202.         break;
  1203.     case rtfRPosPageH:
  1204.         break;
  1205.     case rtfRPosColH:
  1206.         break;
  1207.     }
  1208. }
  1209.  
  1210. InStateStack *SaveIState(){
  1211.     InStateStack *NewState;
  1212.     NewState=(InStateStack *) malloc(
  1213.         (int) sizeof(InStateStack));
  1214.         if(NewState==NULL){
  1215.         RTFPanic ("Memory Allocation Failed");
  1216.     }
  1217.     NewState->ParStyle=ParStyle;
  1218.     NewState->TStyle=TStyle;
  1219.     NewState->TFont=TFont;
  1220.     NewState->TSize=TSize;
  1221.     NewState->destination=destination;
  1222.     NewState->firstcell=firstcell;
  1223.     NewState->inTable=inTable;
  1224.     NewState->cellno=cellno;
  1225.     NewState->lastcell=lastcell;
  1226.     NewState->ToCLev=ToCLev;
  1227.     IStyle_Chg=1;    
  1228.     return(NewState);
  1229. }
  1230. PushIState(){
  1231. /*
  1232.     Push the current state. Note that defaults are not set,
  1233.     so resetting the default state (such as for headers/footers)
  1234.     must be done after the Push.
  1235. */
  1236.     InStateStack *Current=ISS;
  1237.     ISS=SaveIState();
  1238.     ISS->Next=Current;
  1239. }
  1240. void
  1241. RestoreIState(TheState)
  1242. InStateStack *TheState;
  1243. {
  1244.     ParStyle=TheState->ParStyle;
  1245.     TStyle=TheState->TStyle;
  1246.     TFont=TheState->TFont;
  1247.     TSize=TheState->TSize;
  1248.     ToCLev=TheState->ToCLev;
  1249.     if(destination==rtfFootnote){
  1250.         /* Footnotes get their own tables */
  1251.         firstcell=TheState->firstcell;
  1252.         inTable=TheState->inTable;
  1253.         cellno=TheState->cellno;
  1254.         lastcell=TheState->lastcell;
  1255.     }
  1256.     destination=TheState->destination;
  1257.     RTFFree(TheState);
  1258.     IStyle_Chg=1;
  1259. }
  1260.  
  1261. PopIState(){
  1262.     InStateStack *Next;
  1263.     if(ISS==NULL){
  1264.         RTFPanic ("Input State Stack Popped too many times");
  1265.     }
  1266.     HTMLEndDivert();
  1267.     Next=ISS->Next;
  1268.     RestoreIState(ISS);
  1269.     ISS=Next;
  1270. }
  1271.